package com.king.framework.aspectj;

import java.util.Collection;
import java.util.Date;
import java.util.Map;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.validation.BindingResult;
import org.springframework.web.multipart.MultipartFile;

import com.alibaba.fastjson2.JSON;
import com.king.common.annotation.Log;
import com.king.common.enums.HttpMethod;
import com.king.common.enums.SsEnum;
import com.king.common.utils.SecurityUtils;
import com.king.common.utils.ServletUtils;
import com.king.common.utils.ip.IpUtils;
import com.king.common.utils.text.StringUtils;
import com.king.framework.base.domain.vo.LoginUser;
import com.king.framework.filter.PropertyPreExcludeFilter;
import com.king.framework.manager.AsyncManager;
import com.king.framework.manager.factory.AsyncFactory;
import com.king.modules.monitor.domain.SysOperLogEntity;

/**
 * 操作日志记录处理
 * 
 * @author king
 */
//@Aspect
//@Component
public class LogAspect {
	private static final Logger log = LoggerFactory.getLogger(LogAspect.class);

	/** 排除敏感属性字段 */
	public static final String[] EXCLUDE_PROPERTIES = { "password", "oldPassword", "newPassword", "confirmPassword" };

	/**
	 * 处理完请求后执行
	 *
	 * @param joinPoint
	 *            切点
	 */
	@AfterReturning(pointcut = "@annotation(controllerLog)", returning = "jsonResult")
	public void doAfterReturning(JoinPoint joinPoint, Log controllerLog, Object jsonResult) {
		handleLog(joinPoint, controllerLog, null, jsonResult);
	}

	/**
	 * 拦截异常操作
	 * 
	 * @param joinPoint
	 *            切点
	 * @param e
	 *            异常
	 */
	@AfterThrowing(value = "@annotation(controllerLog)", throwing = "e")
	public void doAfterThrowing(JoinPoint joinPoint, Log controllerLog, Exception e) {
		handleLog(joinPoint, controllerLog, e, null);
	}

	protected void handleLog(final JoinPoint joinPoint, Log controllerLog, final Exception e, Object jsonResult) {
		try {
			// 获取当前的用户
			LoginUser loginUser = SecurityUtils.getLoginUser();

			// *========数据库日志=========*//
			SysOperLogEntity operLog = new SysOperLogEntity();
			operLog.setStatus(SsEnum.NORMAL.getKey());
			// 请求的地址
			String ip = IpUtils.getIpAddr(ServletUtils.getRequest());
			operLog.setOperIp(ip);
			operLog.setOperUrl(StringUtils.substring(ServletUtils.getRequest().getRequestURI(), 0, 255));
			if (loginUser != null) {
				operLog.setOperName(loginUser.getUsername());
				operLog.setGroupid(loginUser.getGroupid());
				operLog.setCr(loginUser.getUserId());
				operLog.setMr(loginUser.getUserId());
			}
			Date ts = new Date();
			operLog.setSs(SsEnum.NORMAL.getKey());
			operLog.setCts(ts);
			operLog.setMts(ts);
			

			if (e != null) {
				operLog.setStatus(SsEnum.DELETED.getKey());
				operLog.setErrorMsg(StringUtils.substring(e.getMessage(), 0, 2000));
			}
			// 设置方法名称
			String className = joinPoint.getTarget().getClass().getName();
			String methodName = joinPoint.getSignature().getName();
			operLog.setMethod(className + "." + methodName + "()");
			// 设置请求方式
			operLog.setRequestMethod(ServletUtils.getRequest().getMethod());
			// 处理设置注解上的参数
			getControllerMethodDescription(joinPoint, controllerLog, operLog, jsonResult);
			// 保存数据库
			AsyncManager.me().execute(AsyncFactory.recordOperLog(operLog));
		} catch (Exception exp) {
			// 记录本地异常日志
			log.error("异常信息:{}", exp.getMessage());
			exp.printStackTrace();
		}
	}

	/**
	 * 获取注解中对方法的描述信息 用于Controller层注解
	 * 
	 * @param log
	 *            日志
	 * @param operLog
	 *            操作日志
	 * @throws Exception
	 */
	public void getControllerMethodDescription(JoinPoint joinPoint, Log log, SysOperLogEntity operLog, Object jsonResult)
			throws Exception {
		// 设置action动作
		operLog.setBusinessType(log.businessType().ordinal());
		// 设置标题
		operLog.setTitle(log.title());
		// 设置操作人类别
		operLog.setOperatorType(log.operatorType().ordinal());
		// 是否需要保存request，参数和值
		if (log.isSaveRequestData()) {
			// 获取参数的信息，传入到数据库中。
			setRequestValue(joinPoint, operLog);
		}
		// 是否需要保存response，参数和值
		if (log.isSaveResponseData() && StringUtils.isNotNull(jsonResult)) {
			operLog.setJsonResult(StringUtils.substring(JSON.toJSONString(jsonResult), 0, 2000));
		}
	}

	/**
	 * 获取请求的参数，放到log中
	 * 
	 * @param operLog
	 *            操作日志
	 * @throws Exception
	 *             异常
	 */
	private void setRequestValue(JoinPoint joinPoint, SysOperLogEntity operLog) throws Exception {
		String requestMethod = operLog.getRequestMethod();
		if (HttpMethod.PUT.name().equals(requestMethod) || HttpMethod.POST.name().equals(requestMethod)) {
			String params = argsArrayToString(joinPoint.getArgs());
			operLog.setOperParam(StringUtils.substring(params, 0, 2000));
		} else {
			Map<?, ?> paramsMap = ServletUtils.getParamMap(ServletUtils.getRequest());
			operLog.setOperParam(
					StringUtils.substring(JSON.toJSONString(paramsMap, excludePropertyPreFilter()), 0, 2000));
		}
	}

	/**
	 * 参数拼装
	 */
	private String argsArrayToString(Object[] paramsArray) {
		String params = "";
		if (paramsArray != null && paramsArray.length > 0) {
			for (Object o : paramsArray) {
				if (StringUtils.isNotNull(o) && !isFilterObject(o)) {
					try {
						String jsonObj = JSON.toJSONString(o, excludePropertyPreFilter());
						params += jsonObj.toString() + " ";
					} catch (Exception e) {
					}
				}
			}
		}
		return params.trim();
	}

	/**
	 * 忽略敏感属性
	 */
	public PropertyPreExcludeFilter excludePropertyPreFilter() {
		return new PropertyPreExcludeFilter().addExcludes(EXCLUDE_PROPERTIES);
	}

	/**
	 * 判断是否需要过滤的对象。
	 * 
	 * @param o
	 *            对象信息。
	 * @return 如果是需要过滤的对象，则返回true；否则返回false。
	 */
	@SuppressWarnings("rawtypes")
	public boolean isFilterObject(final Object o) {
		Class<?> clazz = o.getClass();
		if (clazz.isArray()) {
			return clazz.getComponentType().isAssignableFrom(MultipartFile.class);
		} else if (Collection.class.isAssignableFrom(clazz)) {
			Collection collection = (Collection) o;
			for (Object value : collection) {
				return value instanceof MultipartFile;
			}
		} else if (Map.class.isAssignableFrom(clazz)) {
			Map map = (Map) o;
			for (Object value : map.entrySet()) {
				Map.Entry entry = (Map.Entry) value;
				return entry.getValue() instanceof MultipartFile;
			}
		}
		return o instanceof MultipartFile || o instanceof HttpServletRequest || o instanceof HttpServletResponse
				|| o instanceof BindingResult;
	}
}
